home *** CD-ROM | disk | FTP | other *** search
/ Linux Cubed Series 8: LINUX Games / Linux Cubed Series 8 - LINUX Games.iso / games / x11 / strategy / xpat2-1.000 / xpat2-1 / xpat2-1.04 / src / messages.c < prev    next >
C/C++ Source or Header  |  1995-01-28  |  10KB  |  381 lines

  1. /*****************************************************************************/
  2. /*                                         */
  3. /*                                         */
  4. /*    X patience version 2 -- module messages.c                 */
  5. /*                                         */
  6. /*    Internationalisation and keyboard translation/customisation         */
  7. /*    written by Michael Bischoff (mbi@mo.math.nat.tu-bs.de)             */
  8. /*    04-Apr-1994                                 */
  9. /*    see COPYRIGHT.xpat2 for Copyright details                 */
  10. /*                                         */
  11. /*                                         */
  12. /*****************************************************************************/
  13. #include "xpatgame.h"
  14.  
  15. static const char
  16.     n_ace[] = "Ace", n_deuce[] = "Deuce", n_three[] = "Three",
  17.     n_four[] = "Four", n_five[] = "Five", n_six[] = "Six", n_seven[] = "Seven",
  18.     n_eight[] = "Eight", n_nine[] = "Nine", n_ten[] = "Ten", n_jack[] = "Jack",
  19.     n_queen[] = "Queen", n_king[] = "King", n_joker[] = "Joker",
  20.     n_diamonds[] = "Diamonds", n_hearts[] = "Hearts", n_spades[] = "Spades",
  21.     n_clubs[] = "Clubs";
  22.  
  23. const char *US_rank_name[] = {
  24.     n_ace, n_deuce, n_three, n_four, n_five, n_six, n_seven, n_eight, n_nine,
  25.     n_ten, n_jack, n_queen, n_king };
  26. const char *US_suit_name[] = {
  27.     n_clubs, n_spades, n_hearts, n_diamonds };
  28.  
  29. const char *xpat_messages[] = {
  30.     "Quit game?",
  31.     "Another game?",
  32.     "Restart game?",
  33.     n_clubs,
  34.     n_spades,
  35.     n_hearts,
  36.     n_diamonds,
  37.     n_joker,
  38.     n_ace,
  39.     n_deuce,
  40.     n_three,
  41.     n_four,
  42.     n_five,
  43.     n_six,
  44.     n_seven,
  45.     n_eight,
  46.     n_nine,
  47.     n_ten,
  48.     n_jack,
  49.     n_queen,
  50.     n_king,
  51.     "stack",
  52.     "slot",
  53.     "the talon",
  54.     "tmp",
  55.     "the stock",
  56.     "No hint was given!",
  57.     "You can move %P (%R of %S) to %P.",
  58.     "No hints!",
  59.     "(wrapping)",
  60.     "Bookmark set.",
  61.     "You won!",
  62.     "No moves to stack possible.",
  63.     "Saving of game FAILED.",
  64.     "Loading the game FAILED.",
  65.     "Could not open file.",
  66.     "Could not open file.",
  67.     "Could not write header.",
  68.     "Could not read header.",
  69.     "Could not write moves.",
  70.     "Could not read moves.",
  71.     "Saving the game succeeded.",
  72.     "Loading the game succeeded.",
  73.     "Magic match failed.",
  74.     "No flip left.",
  75.     "flipping",
  76.     "new cards",
  77.     "No new cards!",
  78.     "No source pile selected.",
  79.     "Cards moved to stack.",
  80.     "Move not possible.",
  81.     "Undo not possible.",
  82.     "Undo!",
  83.     "Undo (cheat!)",
  84.     "Redo not possible.",
  85.     "Redo!",
  86.     "Redo (& uncheat)",
  87.     "Invalid move",
  88.     "Empty or invalid source pile",
  89.     "source pile selected",
  90.     "not so many slots",
  91.     "aborted",
  92.     "Unknown command:",
  93.     "no move possible",
  94.     "Xpat2 version",
  95.     "Seed is %ld, cheat count %d. ",
  96.     "Score %d of %d.",
  97.     "(No score for these rules)",
  98.     "",
  99.     "Help on keys",
  100.     "Help on %s rules",
  101.     "help.keys",
  102.     "",
  103.     "",
  104.     "",
  105.     "",
  106.     "help.Spider",    /* we keep the names to 14 characters. Not nice, but POSIX */
  107.     "help.Gypsy",
  108.     "help.Klondike",
  109.     "help.Seahaven",
  110.     "help.FreeCell",    /* 80 */
  111.     "help.Delight",
  112.     "hlpMonteCarlo",
  113.     "hlpMidnightOil",
  114.     "hlpCalculation",
  115.     "hlpmodCanfield",
  116.     "help.Michaels",
  117.     "help.Canfield",
  118.     "help.Cotillion",
  119.     "hlpBakersDozen",
  120.     "",
  121.     "",
  122.     "Please wait, loading cards...",
  123.     "Welcome to XPat2 version %s!",
  124.     "",
  125.     "Rotate up selected.",
  126.     "Rotate down selected.",
  127.     "",
  128.     "Card must be visible.",
  129.     "For this action, a slot is required.",
  130.     "In these rules, the cards cannot be shuffled.",    /* 100 */
  131.     "No shuffles left.",
  132.     "One shuffle left.",
  133.     "%d shuffles left.",
  134.     "In these rules, the deck cannot be flipped.",
  135.     "No flip left.",
  136.     "One flip left.",
  137.     "%d flips left.",
  138.     "This function is not available in these rules.",
  139.     "No slot rotations left.",
  140.     "One slot rotation left.",
  141.     "%d slot rotations left.",
  142.     "No ``deal cards'' in these rules.",
  143.     "No more cards on the deck.",
  144.     "One last card on the deck.",
  145.     "%d cards left on deck.",
  146.     "Starting position.",
  147.     "One move done.",
  148.     "%d moves done.",
  149.     "",
  150.     "Not a single point of",
  151.     "One point of",
  152.     "Score: %d points of",
  153. };
  154.  
  155. void *malloc_(size_t n) {
  156.     void *p;
  157.     if (!(p = malloc(n))) {
  158.     fprintf(stderr, "out of memory\n");
  159.     exit(EXIT_FAILURE);
  160.     }
  161.     return p;
  162. }
  163.  
  164. char *strsav(const char *txt) {
  165.     char *p = malloc_(1 + strlen(txt));
  166.     strcpy(p, txt);
  167.     return p;
  168. }
  169.  
  170. void read_message_file(const char *filename) {
  171.     FILE *fp;
  172.     int i;
  173.     char line[256];
  174.     if (*filename != '/') {
  175.     sprintf(line, "%s/%s/%s", LIBDIR, langdir, filename);
  176.     filename = line;
  177.     }
  178.     if (!(fp = fopen(filename, "r")))
  179.     return;        /* Xt philosophy: ignore error */
  180.     for (i = 0; i < sizeof(xpat_messages) / sizeof(const char *); ++i) {
  181.     char *p;
  182.     again:
  183.     if (!fgets(line, sizeof(line), fp))
  184.         break;    /* EOF */
  185.     if (*line == '#')
  186.         goto again;
  187.     if ((p = strrchr(line, '\n')))
  188.         *p = '\0';
  189.     if (*line)    /* empty lines => keep old text */
  190.         xpat_messages[i] = strsav(line);
  191.     }
  192.     fclose(fp);
  193. }
  194.  
  195. static void cmd_None(void) {}
  196.  
  197. static void replace_binding(struct key_action *p, const char *name) {
  198.     int i;
  199.     static struct translator {
  200.     const char *name;
  201.     void (*func)(void);
  202.     } translator[] = {
  203.     { "None",        cmd_None },
  204.     { "rq_LeavePat",    rq_LeavePat },
  205.     { "rq_RestartGame",    rq_RestartGame },
  206.     { "rq_AnotherGame",    rq_AnotherGame },
  207.     { "ResetHints",        cmd_ResetHints },
  208.     { "DoHint",        cmd_DoHint },
  209.     { "NextHint",        cmd_NextHint },
  210.     { "PreviousHint",    cmd_PreviousHint },
  211.     { "CancelSelection",    cmd_CancelSelection },
  212.     { "AllMoves",        cmd_AllMoves },
  213.     { "UndoMove",        cmd_UndoMove },
  214.     { "RedoMove",        cmd_RedoMove },
  215.     { "DealCards",        cmd_DealCards },
  216.     { "LeavePat",        cmd_LeavePat },
  217.     { "ShowScore",        cmd_ShowScore },
  218.     { "AnotherGame",    cmd_AnotherGame },
  219.     { "RestartGame",    cmd_RestartGame },
  220.     { "ReplayGame",        cmd_ReplayGame },
  221.     { "Layout",        cmd_Layout },
  222.     { "SaveGame",        cmd_SaveGame },
  223.     { "ShowVersion",    cmd_ShowVersion },
  224.     { "MinWindow",        cmd_MinWindow },
  225.     { "PreferredWindow",    cmd_PreferredWindow },
  226.     { "Info",        cmd_Info },
  227.     { "ToStack",        cmd_ToStack },
  228.     { "OneToStack",        cmd_OneToStack },    /* selected card(s) */
  229.     { "AllToStack",        cmd_AllToStack },    /* all possible */
  230.     { "DropBookmark",    cmd_DropBookmark },
  231.     { "GotoBookmark",    cmd_GotoBookmark },
  232.     { "RotateUp",        cmd_RotateUp },
  233.     { "RotateDown",        cmd_RotateDown },
  234.     { "Cancel",        cmd_Cancel },
  235.     { "Confirm",        cmd_Confirm }
  236.     };
  237.     for (i = 0; i < sizeof(translator) / sizeof(translator[0]); ++i)
  238.     if (!strcmp(name, translator[i].name)) {
  239.         p->action = translator[i].func;
  240.         return;
  241.     }
  242.     fprintf(stderr, "WARNING: no function corresponds to \"%s\"\n", name);
  243.     p->action = cmd_None;
  244. }
  245.  
  246. static struct key_action *global_bindings = NULL;
  247. const char *langdir = "";
  248.  
  249. void add_keybinding(struct key_action **cp, const char *cmd, const char *function) {
  250.     /* a NULL pointer for cmd is a wildcard */
  251.     int done = 0;
  252.     struct key_action *p;
  253.  
  254.     if (!cp)
  255.     return;        /* bindings for unimplemented rulesets */
  256.     while (*cp) {
  257.     p = *cp;
  258.     if (!cmd || !strcmp(p->string, cmd)) {
  259.         /* replace this! */
  260.         replace_binding(p, function);
  261.         done = 1;
  262.     }
  263.     cp = &(p->next);
  264.     }
  265.     if (!done && cmd) {
  266.     /* didn't find previous command */
  267.     /* add a new entry */
  268.     p = *cp = malloc_(sizeof(struct key_action));
  269.     p->next = NULL;
  270.     p->string = strsav(cmd);
  271.     replace_binding(p, function);
  272.     }
  273. }
  274.  
  275.  
  276. void read_keyboard_file(const char *filename) {
  277.     FILE *fp;
  278.     char line[256];
  279.     char buff[32], cmd[2];
  280.     struct key_action **cp = &global_bindings;
  281.  
  282.     cmd[1] = '\0';    /* 1-char commands currently */
  283.     if (*filename != '/') {
  284.     sprintf(line, "%s/%s/%s", LIBDIR, langdir, filename);
  285.     filename = line;
  286.     }
  287.     if (!(fp = fopen(filename, "r"))) {
  288.     fprintf(stderr, "FAILED to open keyboard file \"%s\"\n", filename);
  289.     return;        /* Xt philosophy: ignore error */
  290.     }
  291.     while (fgets(line, sizeof(line), fp)) {
  292.     char *p;
  293.     if ((p = strrchr(line, '\n')))
  294.         *p = '\0';
  295.     if (!*line)
  296.         continue;
  297.     if (!strncmp(line, "#c", 2))    /* comment */
  298.         continue;
  299.     if (!strncmp(line, "#rules ", 7)) {    /* switch to other rules */
  300.         struct rules *rp;
  301.         p = line+7;
  302.         if (!(rp = getrules(p)))
  303.         cp = NULL;
  304.         else
  305.         cp = &(rp->local_bindings);
  306.         continue;
  307.     }
  308.     if (!strncmp(line, "#include ", 9)) {    /* include */
  309.         read_keyboard_file(line+9);
  310.         continue;
  311.     }
  312.     if (!strncmp(line, "#x", 2) && strlen(line) < 32) { /* hex number */
  313.         int c;
  314.         sscanf(line+2, "%x %s", &c, buff);
  315.         cmd[0] = c;
  316.         add_keybinding(cp, cmd, buff);
  317.         continue;
  318.     }
  319.     /* else assume string<space>commandname */
  320.     if (strlen(line) < 32 && (p = strchr(line+1, ' '))) {
  321.         *p = '\0';
  322.         add_keybinding(cp, line, p+1);
  323.         continue;
  324.     }
  325.     fprintf(stderr, "Warning: cannot parse line in keys file:%s\n", line);
  326.     }
  327.     fclose(fp);
  328. }
  329.  
  330.  
  331. void key_pressed(char *str) {
  332.     struct key_action *p;
  333.  
  334.     /* search for local binding */
  335.     for (p = rules.local_bindings; p; p = p->next)
  336.     if (!strcmp(p->string, str)) {
  337.         (*p->action)();
  338.         return;
  339.     }
  340.  
  341.     /* search for global binding */
  342.     for (p = global_bindings; p; p = p->next)
  343.     if (!strcmp(p->string, str)) {
  344.         (*p->action)();
  345.         return;
  346.     }
  347.  
  348.     /* not found. break up the string to smaller pieces */
  349.     if (strlen(str) > 1) {
  350.     char s[2];
  351.     while (*str) {
  352.         s[0] = *str++;
  353.         s[1] = '\0';
  354.         key_pressed(s);
  355.     }
  356.     } else    /* last resort: */
  357.     /* use hardcoded entries. Note: These can be overridden by assigning them "None" before. */
  358.  
  359.     switch (str[0]) {
  360.     case '\014':
  361.     cmd_ResetHints();    /* complicated to save */
  362.     refresh_screen();
  363.     break;
  364.     case 27:    /* abort cmd */
  365.     cmd_CancelSelection();
  366.     show_message(TXT_ABORTED);
  367.     return;
  368.     }
  369.     if (str[0]) {
  370.     const char *rulechange = "FGHKSIMOBCR";
  371.     const char *rulename[] = { "Free Cell", "Gypsy", "Seahaven", "Klondike",
  372.                        "Spider", "id", "mc", "mo", "bd", "Canfield", "rc" };
  373.     const char *s;
  374.  
  375.     if ((s = strchr(rulechange, str[0]))) {
  376.         change_rules(rulename[s-rulechange]);
  377.         return;
  378.     }
  379.     }
  380. }
  381.